@[TOC](Spring Cloud Gateway)
1. 简介
SpringCloud Gateway 旨在为微服务架构提供一种简单有效的统一的 API 路由管理方式。
SpringCloud Gateway 作为 Spring Cloud 生态系统中的网关,目标是替代 Zuul,使用的是Spring Boot和Spring Webflux提供的Netty底层环境,不能和传统的Servlet容器一起使用,也不能打包成一个WAR包。
SpringCloud官 对SpringCloud Gateway 特征介绍如下:
(1)基于 Spring Framework 5,Project Reactor 和 Spring Boot 2.0
(2)集成 Hystrix 断路器
(3)集成 Spring Cloud DiscoveryClient
(4)Predicates 和 Filters 作用于特定路由,易于编写的 Predicates 和 Filters
(5)具备一些网关的高级功能:动态路由、限流、路径重写
从以上的特征来说,和Zuul的特征差别不大。SpringCloud Gateway和Zuul主要的区别,还是在底层的通信框架上。
简单说明一下上文中的三个术语:
(1)Filter(过滤器):
和Zuul的过滤器在概念上类似,可以使用它拦截和修改请求,并且对上游的响应,进行二次处理。过滤器为org.springframework.cloud.gateway.filter.GatewayFilter类的实例。
(2)Route(路由):
网关配置的基本组成模块,和Zuul的路由配置模块类似。一个Route模块由一个 ID,一个目标 URI,一组断言和一组过滤器定义。如果断言为真,则路由匹配,目标URI会被访问。
(3)Predicate(断言):
这是一个 Java 8 的 Predicate,可以使用它来匹配来自 HTTP 请求的任何内容,例如 headers 或参数。断言的输入类型是一个 ServerWebExchange。
2. 基本用法
SpringCloud Gateway 支持两种不同的用法:
2.1 首先来看编码式:
创建 Spring Boot 项目,添加依赖:

项目创建成功后,直接配置一个 RouteLocator 这样一个Bean,就可以实现请求转发。
@Bean RouteLocator routeLocator(RouteLocatorBuilder builder){ return builder.routes() .route("javaboy", r -> r.path("/get").uri("http://httpbin.org")) .build(); }
|
这里只需提供 RouteLocator 的Bean 就可实现请求转发,配置完成后,访问 http://localhost:8080/get 就会转发至 http://httpbin.org 接口。
2.2 yml 配置
注释掉上面的Bean ,配置如下yml,重启服务与源来效果相同。
spring: cloud: gateway: routes: - id: java uri: http://httpbin.org predicates: - Path=/get
|
2.3 与微服务结合
还是延用以前的eurek server和provider以及gateway。首先给 gateway 添加依赖:
<dependency> <groupId>org.springframework.cloud</groupId> <artifactId>spring-cloud-starter-netflix-eureka-client</artifactId> </dependency>
|

然后配置yml :
spring: cloud: gateway: discovery: locator: # 开启自动代理 enabled: true # 服务id为true --> 这样小写服务就可访问了 lower-case-service-id: true application: name: gateway
eureka: client: service-url: defaultZone: http://localhost:1111/eureka
#日志 logging: level: org.springframework.cloud.gateway: debug
|
接下来启动eurek server和provider以及gateway,就可以通过gateway (http://localhost:8080/provider/hello)
访问到其他注册到Erueka上的服务了。访问方式和Zuul 类似。
小结:gateway 注册到 eureka ,就可以自动代理注册中心的所有服务,简化路由配置的使用方式。
3. Predicate(断言)
3.1 时间断言
- Before –> 请求时间在某个时间点之前 才会被路由
- After –> 请求时间在某个时间点之后 才会被路由
- Between –> 请求时间在2个时间点之间(两个时间点,分开)
spring: cloud: gateway: routes: - id: before_route uri: http://www.baidu.com # 时间断言 predicates: - Before=2021-03-31T08:38:47+08:00
|
访问: localhost:8080 如下:

如果改为 - Before=2020-03-31T08:38:47+08:00 重启访问后如下:

3.2 请求方法&请求路径断言
spring: cloud: gateway: routes: - id: java_route uri: http://httpbin.org predicates: - Method=GET
|
上面这个配置表示只给 GET请求路由,访问 http://localhost:8080/get 就成功。
spring: cloud: gateway: routes: - id: java_route uri: http://httpbin.org predicates: - Path=/2020/0912/{segment}
|
上面这个配置表示路径满足这个规则都会进行转发,例如:http://www.xxx.com/2020/0912/** ,否则不转发。
3.3 Query 断言
spring: cloud: gateway: routes: - id: java_route uri: http://httpbin.org predicates: - Query=name
|
上面配置表示一定要有 name 参数才转发,否则不会转发。
也可以指定参数的值,例如参数的 key 为 name ,value 必须要以 java 开始。 如下:
spring: cloud: gateway: routes: - id: java_route uri: http://httpbin.org predicates: - Query=name,java.*
|
3.4 多种匹配方式
spring: cloud: gateway: routes: # 组合使用 - id: many_route uri: http://httpbin.org predicates: # 可以指定参数和值 ?name=1开头的值 - Query=name,1.* - Method=GET - After=2020-03-01T08:38:47+08:00
|
满足以上条件即可转发。
3.5 自定义路由断言
配置自定义断言工厂:
@Component public class CheckAuthRoutePredicateFactory extends AbstractRoutePredicateFactory<CheckAuthRoutePredicateFactory.User> { public CheckAuthRoutePredicateFactory() { super(User.class); } @Override public Predicate<ServerWebExchange> apply(User config) { return exchange -> { System.out.println("进入apply:" + config.getName()); if (config.getName().equals("kitty")){ return true; } return false; }; } public static class User{ private String name; public String getName() { return name; } public void setName(String name) { this.name = name; } } }
|
yml 配置:
spring: cloud: gateway: routes: # 自定义断言工厂 -name就是之前以xxxRoutePredicateFactory的xxxx为断言工厂名 - id: customer_route uri: http://httpbin.org predicates: - name: CheckAuth args: name: kitty
|
当参数name=kitty与自定义断言工厂的getName一致时:

当参数name=kitty与自定义断言工厂的getName不一致时

4. Filter(过滤器)
gateway过滤器分类:
- GlobalFilter
- GatewayFilter
这里的过滤器就是在请求转发的时候,自动额外添加参数。
4.1 AddRequestParameter过滤器使用
spring: cloud: gateway: routes: - id: java_route uri: lb://provider filters: - AddRequestParameter=name,javaboy predicates: - Method=GET
|
